home *** CD-ROM | disk | FTP | other *** search
/ SPACE 2 / SPACE - Library 2 - Volume 1.iso / program / 515 / rcs5ap1s.lzh / RCSEDIT.C < prev    next >
C/C++ Source or Header  |  1991-01-30  |  24KB  |  883 lines

  1. /*
  2.  *                     RCS stream editor
  3.  */
  4. /**********************************************************************************
  5.  *                       edits the input file according to a
  6.  *                       script from stdin, generated by diff -n
  7.  *                       performs keyword expansion
  8.  **********************************************************************************
  9.  */
  10.  
  11. /* Copyright (C) 1982, 1988, 1989 Walter Tichy
  12.    Copyright 1990 by Paul Eggert
  13.    Distributed under license by the Free Software Foundation, Inc.
  14.  
  15. This file is part of RCS.
  16.  
  17. RCS is free software; you can redistribute it and/or modify
  18. it under the terms of the GNU General Public License as published by
  19. the Free Software Foundation; either version 1, or (at your option)
  20. any later version.
  21.  
  22. RCS is distributed in the hope that it will be useful,
  23. but WITHOUT ANY WARRANTY; without even the implied warranty of
  24. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  25. GNU General Public License for more details.
  26.  
  27. You should have received a copy of the GNU General Public License
  28. along with RCS; see the file COPYING.  If not, write to
  29. the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
  30.  
  31. Report problems and direct all questions to:
  32.  
  33.     rcs-bugs@cs.purdue.edu
  34.  
  35. */
  36.  
  37.  
  38. /* $Log: rcsedit.c,v $
  39.  * Revision 5.8  1991/01/30  14:21:32  apratt
  40.  * CI with RCS version 5
  41.  *
  42.  * Revision 5.7  91/01/29  17:45:46  apratt
  43.  * Added AKP_BUGFIXES around my bug fixes
  44.  * 
  45.  * Revision 5.6  91/01/16  15:44:50  apratt
  46.  * This version works passably on the ST.
  47.  * 
  48.  * Revision 5.5  90/12/30  05:07:35  eggert
  49.  * checked in with -k by apratt at 91.01.10.13.15.10.
  50.  * 
  51.  * Revision 5.5  1990/12/30  05:07:35  eggert
  52.  * Fix report of busy RCS files when !defined(O_CREAT) | !defined(O_EXCL).
  53.  *
  54.  * Revision 5.4  1990/11/01  05:03:40  eggert
  55.  * Permit arbitrary data in comment leaders.
  56.  *
  57.  * Revision 5.3  1990/09/11  02:41:13  eggert
  58.  * Tune expandline().
  59.  *
  60.  * Revision 5.2  1990/09/04  08:02:21  eggert
  61.  * Count RCS lines better.  Improve incomplete line handling.
  62.  *
  63.  * Revision 5.1  1990/08/29  07:13:56  eggert
  64.  * Add -kkvl.
  65.  * Fix bug when getting revisions to files ending in incomplete lines.
  66.  * Fix bug in comment leader expansion.
  67.  *
  68.  * Revision 5.0  1990/08/22  08:12:47  eggert
  69.  * Don't require final newline.
  70.  * Don't append "checked in with -k by " to logs,
  71.  * so that checking in a program with -k doesn't change it.
  72.  * Don't generate trailing white space for empty comment leader.
  73.  * Remove compile-time limits; use malloc instead.  Add -k, -V.
  74.  * Permit dates past 1999/12/31.  Make lock and temp files faster and safer.
  75.  * Ansify and Posixate.  Check diff's output.
  76.  *
  77.  * Revision 4.8  89/05/01  15:12:35  narten
  78.  * changed copyright header to reflect current distribution rules
  79.  * 
  80.  * Revision 4.7  88/11/08  13:54:14  narten
  81.  * misplaced semicolon caused infinite loop
  82.  * 
  83.  * Revision 4.6  88/08/09  19:12:45  eggert
  84.  * Shrink stdio code size; allow cc -R.
  85.  * 
  86.  * Revision 4.5  87/12/18  11:38:46  narten
  87.  * Changes from the 43. version. Don't know the significance of the
  88.  * first change involving "rewind". Also, additional "lint" cleanup.
  89.  * (Guy Harris)
  90.  * 
  91.  * Revision 4.4  87/10/18  10:32:21  narten
  92.  * Updating version numbers. Changes relative to version 1.1 actually
  93.  * relative to 4.1
  94.  * 
  95.  * Revision 1.4  87/09/24  13:59:29  narten
  96.  * Sources now pass through lint (if you ignore printf/sprintf/fprintf 
  97.  * warnings)
  98.  * 
  99.  * Revision 1.3  87/09/15  16:39:39  shepler
  100.  * added an initializatin of the variables editline and linecorr
  101.  * this will be done each time a file is processed.
  102.  * (there was an obscure bug where if co was used to retrieve multiple files
  103.  *  it would dump)
  104.  * fix attributed to  Roy Morris @FileNet Corp ...!felix!roy
  105.  * 
  106.  * Revision 1.2  87/03/27  14:22:17  jenkins
  107.  * Port to suns
  108.  * 
  109.  * Revision 4.1  83/05/12  13:10:30  wft
  110.  * Added new markers Id and RCSfile; added locker to Header and Id.
  111.  * Overhauled expandline completely() (problem with $01234567890123456789@).
  112.  * Moved trymatch() and marker table to rcskeys.c.
  113.  * 
  114.  * Revision 3.7  83/05/12  13:04:39  wft
  115.  * Added retry to expandline to resume after failed match which ended in $.
  116.  * Fixed truncation problem for $19chars followed by@@.
  117.  * Log no longer expands full path of RCS file.
  118.  * 
  119.  * Revision 3.6  83/05/11  16:06:30  wft
  120.  * added retry to expandline to resume after failed match which ended in $.
  121.  * Fixed truncation problem for $19chars followed by@@.
  122.  * 
  123.  * Revision 3.5  82/12/04  13:20:56  wft
  124.  * Added expansion of keyword Locker.
  125.  *
  126.  * Revision 3.4  82/12/03  12:26:54  wft
  127.  * Added line number correction in case editing does not start at the
  128.  * beginning of the file.
  129.  * Changed keyword expansion to always print a space before closing KDELIM;
  130.  * Expansion for Header shortened.
  131.  *
  132.  * Revision 3.3  82/11/14  14:49:30  wft
  133.  * removed Suffix from keyword expansion. Replaced fclose with ffclose.
  134.  * keyreplace() gets log message from delta, not from curlogmsg.
  135.  * fixed expression overflow in while(c=putc(GETC....
  136.  * checked nil printing.
  137.  *
  138.  * Revision 3.2  82/10/18  21:13:39  wft
  139.  * I added checks for write errors during the co process, and renamed
  140.  * expandstring() to xpandstring().
  141.  *
  142.  * Revision 3.1  82/10/13  15:52:55  wft
  143.  * changed type of result of getc() from char to int.
  144.  * made keyword expansion loop in expandline() portable to machines
  145.  * without sign-extension.
  146.  */
  147.  
  148.  
  149. #include "rcsbase.h"
  150.  
  151. libId(editId, "$Id: rcsedit.c,v 5.8 1991/01/30 14:21:32 apratt Exp $")
  152.  
  153. static void keyreplace P((enum markers,const struct hshentry*,FILE*));
  154.  
  155.  
  156. FILE *fcopy;         /* result file descriptor                */
  157. const char *resultfile;  /* result file name                    */
  158. int locker_expansion;     /* should the locker name be appended to Id val?   */
  159. static FILE *fedit;     /* edit   file descriptor                */
  160. static const char *editfile;     /* edit file                    */
  161. static const char *editdir;  /* edit directory                    */
  162. static unsigned long editline; /*fedit line counter; is always #lines+1     */
  163. static long linecorr; /* #adds - #deletes in each edit run.            */
  164.                /*used to correct editline in case file is not rewound after */
  165.                /* applying one delta                                        */
  166.  
  167. #define DIRTEMPNAMES 3
  168. enum maker {notmade, real, effective};
  169. struct buf dirtfname[DIRTEMPNAMES];        /* unlink these when done */
  170. static volatile enum maker dirtfmaker[DIRTEMPNAMES];    /* if these are set */
  171.  
  172.     FILE *
  173. initeditfiles(dir)
  174.     const char *dir;
  175. /* Function: Initializes resultfile and editfile with temporary filenames
  176.  * in directory dir. Opens resultfile for reading and writing, with fcopy
  177.  * as file descriptor. fedit is set to nil.
  178.  */
  179. {
  180.     editline = 0;    /* make sure we start from the beginning*/
  181.     linecorr = 0;
  182.     editdir = dir;
  183.     resultfile = makedirtemp(dir,1);
  184.     editfile = nil;
  185.         fedit=nil;
  186.     errno = 0;
  187.     return fcopy = fopen(resultfile,"w+");
  188. }
  189.  
  190.     void
  191. inittmpeditfiles()
  192. {
  193.     if (!initeditfiles(tmp()))
  194.         efaterror(resultfile);
  195. }
  196.  
  197.  
  198.     void
  199. arewind(f)
  200.     FILE *f;
  201. {
  202.     if (fseek(f, (long)0, SEEK_SET) == EOF)
  203.         IOerror();
  204. }
  205.  
  206.     void
  207. swapeditfiles(tostdout)
  208.     int tostdout;
  209. /* Function: swaps resultfile and editfile, assigns fedit=fcopy,
  210.  * rewinds fedit for reading, and opens resultfile for reading and
  211.  * writing, using fcopy. If tostdout, fcopy is set to stdout.
  212.  */
  213. {
  214.     const char *tmpptr;
  215.         fedit=fcopy;
  216.     arewind(fedit);
  217.         editline = 1; linecorr=0;
  218.         tmpptr=editfile; editfile=resultfile; resultfile=tmpptr;
  219.         if (tostdout)
  220.                 fcopy=stdout;
  221.     else {
  222.         if (!resultfile)
  223.         resultfile = makedirtemp(editdir,2);
  224.         errno = 0;
  225.         if (!(fcopy = fopen(resultfile,"w+")))
  226.         efaterror(resultfile);
  227.         }
  228. }
  229.  
  230.  
  231.     void
  232. finishedit(delta)
  233.     const struct hshentry *delta;
  234. /* copy the rest of the edit file and close it (if it exists).
  235.  * if delta!=nil, perform keyword substitution at the same time.
  236.  */
  237. {
  238.     register FILE *fe, *fc;
  239.  
  240.     fe = fedit;
  241.     if (fe) {
  242.         fc = fcopy;
  243.                 if (delta!=nil) {
  244.             while (0 < expandline(fe,fc,delta,false,(FILE*)NULL))
  245.                 ;
  246.                 } else {
  247.             fastcopy(fe,fc);
  248.                 }
  249.         ffclose(fe);
  250.         }
  251. }
  252.  
  253.  
  254.  
  255.     static exiting void
  256. editEndsPrematurely()
  257. {
  258.     fatserror("edit script ends prematurely");
  259. }
  260.  
  261.     static exiting void
  262. editLineNumberOverflow()
  263. {
  264.     fatserror("edit script refers to line past end of file");
  265. }
  266.  
  267.  
  268.     static void
  269. copylines(upto,delta)
  270.     register unsigned long upto;
  271.     const struct hshentry *delta;
  272. /* Function: copies input lines editline..upto-1 from fedit to fcopy.
  273.  * If delta != nil, keyword expansion is done simultaneously.
  274.  * editline is updated. Rewinds a file only if necessary.
  275.  */
  276. {
  277.     register int c;
  278.     register FILE *fe, *fc;
  279.  
  280.     if (upto < editline) {
  281.                 /* swap files */
  282.                 finishedit((struct hshentry *)nil); swapeditfiles(false);
  283.                 /* assumes edit only during last pass, from the beginning*/
  284.         }
  285.     fe = fedit;
  286.     fc = fcopy;
  287.     if (editline < upto)
  288.         if (delta)
  289.         do {
  290.             if (expandline(fe,fc,delta,false,(FILE*)NULL) <= 0)
  291.                 goto unexpected_EOF;
  292.         } while (++editline < upto);
  293.         else
  294.         do {
  295.             do {
  296.                 c = getc(fe);
  297.                 if (c == EOF)
  298.                     goto unexpected_EOF;
  299.                 aputc(c, fc);
  300.             } while (c != '\n');
  301.         } while (++editline < upto);
  302.     return;
  303.  
  304.     unexpected_EOF:
  305.     editLineNumberOverflow();
  306. }
  307.  
  308.  
  309.  
  310.     void
  311. xpandstring(delta)
  312.     const struct hshentry *delta;
  313. /* Function: Reads a string terminated by SDELIM from finptr and writes it
  314.  * to fcopy. Double SDELIM is replaced with single SDELIM.
  315.  * Keyword expansion is performed with data from delta.
  316.  * If foutptr is nonnull, the string is also copied unchanged to foutptr.
  317.  */
  318. {
  319.     while (0 < expandline(finptr,fcopy,delta,true,foutptr))
  320.         ;
  321. }
  322.  
  323.  
  324.     void
  325. copystring()
  326. /* Function: copies a string terminated with a single SDELIM from finptr to
  327.  * fcopy, replacing all double SDELIM with a single SDELIM.
  328.  * If foutptr is nonnull, the string also copied unchanged to foutptr.
  329.  * editline is set to (number of lines copied)+1.
  330.  * Assumption: next character read is first string character.
  331.  */
  332. {    register c;
  333.     register FILE *fin, *frew, *fcop;
  334.     register int amidline;
  335.  
  336.     fin=finptr; frew=foutptr; fcop=fcopy;
  337.         editline=1;
  338.     amidline = false;
  339.     for (;;) {
  340.         GETC(fin,frew,c);
  341.         switch (c) {
  342.             case EOF:
  343.             unterminatedString();
  344.             /*NOTREACHED*/
  345.             case '\n':
  346.             ++editline;
  347.             ++rcsline;
  348.             amidline = false;
  349.             break;
  350.             case SDELIM:
  351.             GETC(fin,frew,c);
  352.             if (c != SDELIM) {
  353.                 /* end of string */
  354.                 nextc = c;
  355.                 editline += amidline;
  356.                 return;
  357.             }
  358.             /* fall into */
  359.             default:
  360.             amidline = true;
  361.             break;
  362.                 }
  363.         aputc(c,fcop);
  364.         }
  365. }
  366.  
  367.  
  368.  
  369.  
  370.     void
  371. editstring(delta)
  372.     const struct hshentry *delta;
  373. /* Function: reads an edit script from finptr and applies it to
  374.  * file fedit; the result is written to fcopy.
  375.  * If delta!=nil, keyword expansion is performed simultaneously.
  376.  * If foutptr is set, the edit script is also copied verbatim to foutptr.
  377.  * Assumes that all these files are open.
  378.  * If running out of lines in fedit, fedit and fcopy are swapped.
  379.  * resultfile and editfile are the names of the files that go with fcopy
  380.  * and fedit, respectively.
  381.  * Assumes the next input character from finptr is the first character of
  382.  * the edit script. Resets nextc on exit.
  383.  */
  384. {
  385.         int ed; /* editor command */
  386.         register int c;
  387.     register FILE *fin, *frew, *f;
  388.     register unsigned long i;
  389.     unsigned long line_lim;
  390.     struct diffcmd dc;
  391.  
  392.         editline += linecorr; linecorr=0; /*correct line number*/
  393.     frew = foutptr;
  394.     fin = finptr;
  395.     line_lim = ULONG_MAX;
  396.     initdiffcmd(&dc);
  397.     while (0  <=  (ed = getdiffcmd(fin,SDELIM,frew,&dc)))
  398.         if (line_lim <= dc.line1)
  399.             editLineNumberOverflow();
  400.         else if (!ed) {
  401.             copylines(dc.line1, delta);
  402.                         /* skip over unwanted lines */
  403.             i = dc.nlines;
  404.             linecorr -= i;
  405.             editline += i;
  406.             f = fedit;
  407.             do {
  408.                                 /*skip next line*/
  409.                 while ((c=getc(f))!='\n')
  410.                     if (c==EOF) {
  411.                         if (i!=1)
  412.                         editLineNumberOverflow();
  413.                         line_lim = dc.dafter;
  414.                         break;
  415.                     }
  416.             } while (--i);
  417.         } else {
  418.             copylines(dc.line1+1, delta); /*copy only; no delete*/
  419.             i = dc.nlines;
  420.             linecorr += i;
  421.             f = fcopy;
  422.             do {
  423.                                 /*copy next line from script*/
  424.                                 if (delta!=nil)
  425.                     switch (expandline(fin,f,delta,true,frew)) {
  426.                     case 0:
  427.                         if (i==1)
  428.                         return;
  429.                         /* fall into */
  430.                     case -1:
  431.                         editEndsPrematurely();
  432.                     }
  433.                                 else {
  434.                        for (;;) {
  435.                         GETC(fin,frew,c);
  436.                         if (c == EOF)
  437.                         editEndsPrematurely();
  438.                         aputc(c, f);
  439.                         if (c == '\n')
  440.                         break;
  441.                         if (c==SDELIM) {
  442.                         GETC(fin,frew,c);
  443.                         if (c!=SDELIM) {
  444.                             if (--i)
  445.                             editEndsPrematurely();
  446.                             nextc = c;
  447.                             return;
  448.                         }
  449.                        }
  450.                        }
  451.                        ++rcsline;
  452.                 }
  453.             } while (--i);
  454.                 }
  455.     GETC(fin,frew,c);
  456.     nextc = c;
  457. }
  458.  
  459.  
  460.  
  461. /* The rest is for keyword expansion */
  462.  
  463.  
  464.  
  465.     int
  466. expandline(in, out, delta, delimstuffed, frew)
  467.     register FILE *in, *out;
  468.     const struct hshentry *delta;
  469.     int delimstuffed;
  470.     register FILE *frew;
  471. /* Function: Reads a line from in and writes it to out.
  472.  * If DELIMSTUFFED is set, double SDELIM is replaced with single SDELIM.
  473.  * Keyword expansion is performed with data from delta.
  474.  * If FREW is set, the line is also copied unchanged to FREW.
  475.  * DELIMSTUFFED must be set if FREW is set.
  476.  * Yields -1 if no data is copied, 0 if an incomplete line is copied,
  477.  * 1 if a complete line is copied.
  478.  */
  479. {
  480.     register c;
  481.     register char * tp;
  482.     register int ds, r;
  483.     const char *tlim;
  484.     char keystring[keylength+2];
  485.     static struct buf keyval;
  486.         enum markers matchresult;
  487.  
  488.     ds = delimstuffed;
  489.     r = -1;
  490.     GETC(in,frew,c);
  491.         for (;;) {
  492.         switch (c) {
  493.             case EOF:
  494.                         if(ds) {
  495.                 unterminatedString();
  496.                         }
  497.             return r;
  498.  
  499.             case SDELIM:
  500.             if (ds) {
  501.                 GETC(in,frew,c);
  502.                 if (c != SDELIM) {
  503.                                 /* end of string */
  504.                                 nextc=c;
  505.                 return r;
  506.                 }
  507.             }
  508.             /* fall into */
  509.             default:
  510.             r = 0;
  511.             aputc(c,out);
  512.             break;
  513.  
  514.             case '\n':
  515.             rcsline += ds;
  516.             aputc(c,out);
  517.             return 1;
  518.  
  519.             case KDELIM:
  520.             r = 0;
  521.                         /* check for keyword */
  522.                         /* first, copy a long enough string into keystring */
  523.             tp=keystring;
  524.             for (;;) {
  525.                 GETC(in,frew,c);
  526.                 if (tp < keystring+keylength)
  527.                 switch (ctab[c]) {
  528.                     case LETTER: case Letter:
  529.                     *tp++ = c;
  530.                     continue;
  531.                     default:
  532.                     break;
  533.                 }
  534.                 break;
  535.                         }
  536.             *tp++ = c; *tp = '\0';
  537.             matchresult = trymatch(keystring);
  538.             if (matchresult==Nomatch) {
  539.                 tp[-1] = 0;
  540.                 aprintf(out, "%c%s", KDELIM, keystring);
  541.                 continue;   /* last c handled properly */
  542.             }
  543.  
  544.             /* Now we have a keyword terminated with a K/VDELIM */
  545.             if (c==VDELIM) {
  546.                   /* try to find closing KDELIM, and replace value */
  547.                   bufalloc(&keyval, 1);
  548.                   tp = keyval.string;
  549.                   tlim = tp + keyval.size;
  550.                   for (;;) {
  551.                       GETC(in,frew,c);
  552.                       if (c==EOF || c=='\n' || c==KDELIM)
  553.                     break;
  554.                       *tp++ =c;
  555.                       if (tlim <= tp)
  556.                       tp = bufenlarge(&keyval, &tlim);
  557.                       if (c==SDELIM && ds) { /*skip next SDELIM */
  558.                         GETC(in,frew,c);
  559.                         if (c != SDELIM) {
  560.                             /* end of string before closing KDELIM or newline */
  561.                             *tp = 0;
  562.                             aprintf(out, "%c%s%s", KDELIM, keystring, keyval.string);
  563.                             nextc = c;
  564.                             return 0;
  565.                         }
  566.                       }
  567.                   }
  568.                   if (c!=KDELIM) {
  569.                     /* couldn't find closing KDELIM -- give up */
  570.                     *tp = 0;
  571.                     aprintf(out, "%c%s%s", KDELIM, keystring, keyval.string);
  572.                     continue;   /* last c handled properly */
  573.                   }
  574.             }
  575.             /* now put out the new keyword value */
  576.             keyreplace(matchresult,delta,out);
  577.                 }
  578.         GETC(in,frew,c);
  579.         }
  580. }
  581.  
  582.  
  583. const char ciklog[ciklogsize] = "checked in with -k by ";
  584.  
  585.     static void
  586. keyreplace(marker,delta,out)
  587.     enum markers marker;
  588.     register const struct hshentry *delta;
  589.     register FILE *out;
  590. /* function: outputs the keyword value(s) corresponding to marker.
  591.  * Attributes are derived from delta.
  592.  */
  593. {
  594.     register const char *sp, *cp, *date;
  595.     register char c;
  596.     register size_t cs, cw, ls;
  597.     int RCSv;
  598.  
  599.     sp = Keyword[(int)marker];
  600.  
  601.     if (Expand == KEY_EXPAND) {
  602.         aprintf(out, "%c%s%c", KDELIM, sp, KDELIM);
  603.         return;
  604.     }
  605.  
  606.         date= delta->date;
  607.     RCSv = RCSversion;
  608.  
  609.     if (Expand == KEYVAL_EXPAND  ||  Expand == KEYVALLOCK_EXPAND)
  610.         aprintf(out, "%c%s%c%c", KDELIM, sp, VDELIM,
  611.             marker==Log && RCSv<VERSION(5)  ?  '\t'  :  ' '
  612.         );
  613.  
  614.         switch (marker) {
  615.         case Author:
  616.         aputs(delta->author, out);
  617.                 break;
  618.         case Date:
  619.         printdate(out, date, " ");
  620.                 break;
  621.         case Id:
  622.     case Header:
  623.         aprintf(out, "%s %s ",
  624.               marker==Id || RCSv<VERSION(4)
  625.             ? bindex(RCSfilename,SLASH)
  626.             : getfullRCSname(),
  627.             delta->num
  628.         );
  629.         printdate(out, date, " ");
  630.         aprintf(out, " %s %s",
  631.             delta->author,
  632.               RCSv==VERSION(3) && delta->lockedby ? "Locked"
  633.             : delta->state
  634.         );
  635.         if (delta->lockedby!=nil)
  636.             if (VERSION(5) <= RCSv) {
  637.             if (locker_expansion || Expand==KEYVALLOCK_EXPAND)
  638.                 aprintf(out, " %s", delta->lockedby);
  639.             } else if (RCSv == VERSION(4))
  640.             aprintf(out, " Locker: %s", delta->lockedby);
  641.                 break;
  642.         case Locker:
  643.         if (delta->lockedby)
  644.             if (
  645.                 locker_expansion
  646.             ||    Expand == KEYVALLOCK_EXPAND
  647.             ||    RCSv <= VERSION(4)
  648.             )
  649.             aputs(delta->lockedby, out);
  650.                 break;
  651.         case Log:
  652.         case RCSfile:
  653.         aputs(bindex(RCSfilename,SLASH), out);
  654.                 break;
  655.         case Revision:
  656.         aputs(delta->num, out);
  657.                 break;
  658.         case Source:
  659.         aputs(getfullRCSname(), out);
  660.                 break;
  661.         case State:
  662.         aputs(delta->state, out);
  663.                 break;
  664.     default:
  665.         break;
  666.         }
  667.     if (Expand == KEYVAL_EXPAND  ||  Expand == KEYVALLOCK_EXPAND) {
  668.         afputc(' ', out);
  669.         afputc(KDELIM, out);
  670.     }
  671.     if (marker == Log) {
  672.         sp = delta->log.string;
  673.         ls = delta->log.size;
  674.         if (sizeof(ciklog)-1<=ls && !strncmp(sp,ciklog,sizeof(ciklog)-1))
  675.             return;
  676.         afputc('\n', out);
  677.         cp = Comment.string;
  678.         cw = cs = Comment.size;
  679.         awrite(cp, cs, out);
  680.         aprintf(out, "Revision %s  ", delta->num);
  681.         printdate(out, date, "  ");
  682.         aprintf(out, "  %s", delta->author);
  683.         /* Do not include state: it may change and is not updated.  */
  684.         /* Comment is the comment leader.  */
  685.         if (VERSION(5) <= RCSv)
  686.             for (;  cw && (cp[cw-1]==' ' || cp[cw-1]=='\t');  --cw)
  687.             ;
  688.         for (;;) {
  689.             afputc('\n', out);
  690.             awrite(cp, cw, out);
  691.             if (!ls)
  692.             break;
  693.             --ls;
  694.             c = *sp++;
  695.             if (c != '\n') {
  696.             awrite(cp+cw, cs-cw, out);
  697.             do {
  698.                 afputc(c,out);
  699.                 if (!ls)
  700.                 break;
  701.                 --ls;
  702.                 c = *sp++;
  703.             } while (c != '\n');
  704.             }
  705.         }
  706.     }
  707. }
  708.  
  709.  
  710.     FILE *
  711. rcswriteopen(RCSname)
  712.     const char *RCSname;
  713. /*
  714.  * Create the lock file corresponding to RCSNAME.
  715.  * Then try to open RCSNAME for reading and yield its FILE* descriptor.
  716.  * If all goes well, discard any previously acquired locks,
  717.  * and set frewrite to the FILE* descriptor of the lock file,
  718.  * which will eventually turn into the new RCS file.
  719.  */
  720. {
  721. #if AKP_BUGFIXES
  722.     int errno_hold;
  723. #endif
  724.     register char *tp;
  725.     register const char *sp, *lp;
  726.     FILE *f;
  727.     int fdesc, r;
  728.     struct buf *dirt;
  729.  
  730.     sp = RCSname;
  731.     dirt = &dirtfname[frewrite != NULL];
  732.     bufalloc(dirt, strlen(sp)+1);
  733.     tp = dirt->string;
  734.     if ((lp = strrchr(sp,SLASH)))
  735.         while (sp<=lp)
  736.             *tp++ = *sp++;
  737.     *tp++ = RCSSEP ? RCSSEP : RCSSUF;
  738. #if RCSSEP
  739.     /* Insert `,' and append file name.  */
  740.     lp = strrchr(sp, RCSSEP);
  741. #else
  742.     /* The file system doesn't allow `,'; use `v' as a poor substitute.  */
  743.     lp = sp + strlen(sp) - 2;
  744. #endif
  745.     while (sp<=lp)
  746.         *tp++ = *sp++;
  747.     *tp = '\0'; /* same length as RCSname */
  748.     tp = dirt->string;
  749.  
  750.     f = NULL;
  751.  
  752.     seteid();
  753.     ignoreints();
  754. #    if !open_can_creat
  755. #        define create(f,m) creat(f, m)
  756. #    else
  757. #        define create(f,m) open(f, O_CREAT|O_WRONLY|o_excltrunc, m)
  758. #        ifdef O_EXCL
  759. #            define o_excltrunc O_EXCL
  760. #        else
  761. #            define o_excltrunc O_TRUNC
  762. #        endif
  763. #    endif
  764. #if AKP_BUGFIXES
  765.     errno = 0;
  766. #endif
  767.     if (0 <= (fdesc = create(tp, S_IRUSR|S_IRGRP|S_IROTH))) {
  768.         dirtfmaker[0] = effective;
  769.         errno = 0;
  770.         f = fopen(RCSname,"r");
  771. #if AKP_BUGFIXES
  772.         errno_hold = errno;
  773. #endif
  774.         if (frewrite)
  775.             /* We already have a lock somewhere else.  */
  776.             if (f) {
  777.             /* Discard the first acquired lock.  */
  778.             ffclose(frewrite);  frewrite = NULL;
  779.             if (unlink(newRCSfilename) < 0) {
  780.                 setrid();
  781.                 efaterror(newRCSfilename);
  782.             }
  783.             bufscpy(&dirtfname[0], tp);
  784.             } else {
  785.             /* Prefer the first acquired lock to this one.  */
  786.             r = close(fdesc)<0 || unlink(tp)<0;
  787.             restoreints();
  788.             setrid();
  789.             if (r)
  790.                 efaterror(tp);
  791. #if AKP_BUGFIXES
  792.             errno = errno_hold;
  793. #endif
  794.             return f;
  795.             }
  796.         if (!(frewrite = fdopen(fdesc,"w"))) {
  797.             setrid();
  798.             efaterror(newRCSfilename);
  799.         }
  800.     }
  801. #if !open_can_creat | !defined(O_EXCL)
  802.     else if (errno != ENOENT) {
  803.         /* Set errno=EEXIST if the RCS file is busy.  */
  804.         struct stat statbuf;
  805.         int old_errno = errno;
  806.         errno  =  stat(tp,&statbuf)==0 ? EEXIST : old_errno;
  807.     }
  808. #endif
  809. #if AKP_BUGFIXES
  810.     errno_hold = errno;
  811. #endif
  812.     restoreints();
  813.     setrid();
  814. #if AKP_BUGFIXES
  815.     errno = errno_hold;
  816. #endif
  817.     return f;
  818. }
  819.  
  820.     void
  821. keepdirtemp(name)
  822.     const char *name;
  823. /* Do not unlink name, either because it's not there any more,
  824.  * or because it has already been unlinked.
  825.  */
  826. {
  827.     register int i;
  828.     for (i=DIRTEMPNAMES; 0<=--i; )
  829.         if (dirtfname[i].string == name) {
  830.             dirtfmaker[i] = notmade;
  831.             return;
  832.         }
  833.     faterror("keepdirtemp");
  834. }
  835.  
  836.     const char *
  837. makedirtemp(name, n)
  838.     register const char *name;
  839.     int n;
  840. /*
  841.  * Have maketemp() do all the work if name==tmp.
  842.  * Otherwise, create a unique filename in name's dir using n and name
  843.  * and store it into the dirtfname[n].
  844.  * Because of storage in tfnames, dirtempunlink() can unlink the file later.
  845.  * Return a pointer to the filename created.
  846.  */
  847. {
  848.     register char *tp;
  849.     register const char *lastslash, *np;
  850.  
  851.     if (name == tmp())
  852.         return maketemp(n);
  853.     bufalloc(&dirtfname[n], strlen(name)+3);
  854.     np = tp = dirtfname[n].string;
  855.     if ((lastslash = strrchr(name,SLASH)))
  856.         while (name<=lastslash)
  857.             *tp++ = *name++;
  858.     *tp++ = RCSSEP ? RCSSEP : RCSSUF;
  859.     *tp++ = 'A'+n;
  860.     while ((*tp++ = *name++))
  861.         ;
  862.     dirtfmaker[n] = real;
  863.     return np;
  864. }
  865.  
  866.     void
  867. dirtempunlink()
  868. /* Clean up makedirtemp() files.  May be invoked by signal handler. */
  869. {
  870.     register int i;
  871.     enum maker m;
  872.  
  873.     for (i = DIRTEMPNAMES;  0 <= --i;  )
  874.         if ((m = dirtfmaker[i]) != notmade) {
  875.         if (m == effective)
  876.             seteid();
  877.         VOID unlink(dirtfname[i].string);
  878.         if (m == effective)
  879.             setrid();
  880.         dirtfmaker[i] = notmade;
  881.         }
  882. }
  883.